home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / MAIL.XPI / bin / chrome / messenger.jar / content / messenger / searchTermOverlay.js < prev    next >
Encoding:
JavaScript  |  2005-07-20  |  17.2 KB  |  514 lines

  1. /* -*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
  2. /* ***** BEGIN LICENSE BLOCK *****
  3.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  4.  *
  5.  * The contents of this file are subject to the Mozilla Public License Version
  6.  * 1.1 (the "License"); you may not use this file except in compliance with
  7.  * the License. You may obtain a copy of the License at
  8.  * http://www.mozilla.org/MPL/
  9.  *
  10.  * Software distributed under the License is distributed on an "AS IS" basis,
  11.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  12.  * for the specific language governing rights and limitations under the
  13.  * License.
  14.  *
  15.  * The Original Code is Mozilla Communicator client code.
  16.  *
  17.  * The Initial Developer of the Original Code is
  18.  * Netscape Communications Corporation.
  19.  * Portions created by the Initial Developer are Copyright (C) 1998
  20.  * the Initial Developer. All Rights Reserved.
  21.  *
  22.  * Contributor(s):
  23.  *   Alec Flett <alecf@netscape.com>
  24.  *   Seth Spitzer <sspitzer@netscape.com>
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either of the GNU General Public License Version 2 or later (the "GPL"),
  28.  * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. var gTotalSearchTerms=0;
  41. var gSearchTermList;
  42. var gSearchTerms = new Array;
  43. var gSearchRemovedTerms = new Array;
  44. var gSearchScope;
  45. var gSearchBooleanRadiogroup;
  46.  
  47. var gUniqueSearchTermCounter = 0; // gets bumped every time we add a search term so we can always 
  48.                                   // dynamically generate unique IDs for the terms.
  49.  
  50. // cache these so we don't have to hit the string bundle for them
  51. var gMoreButtonTooltipText;
  52. var gLessButtonTooltipText;
  53. var gLoading = true;
  54.  
  55.  
  56. function searchTermContainer() {}
  57.  
  58. searchTermContainer.prototype = {
  59.     internalSearchTerm : '',
  60.     internalBooleanAnd : '',
  61.  
  62.     // this.searchTerm: the actual nsIMsgSearchTerm object
  63.     get searchTerm() { return this.internalSearchTerm; },
  64.     set searchTerm(val) {
  65.         this.internalSearchTerm = val;
  66.  
  67.         var term = val;
  68.         // val is a nsIMsgSearchTerm
  69.         var searchAttribute=this.searchattribute;
  70.         var searchOperator=this.searchoperator;
  71.         var searchValue=this.searchvalue;
  72.  
  73.         // now reflect all attributes of the searchterm into the widgets
  74.         if (searchAttribute) searchAttribute.value = term.attrib;
  75.         if (searchOperator) searchOperator.value = val.op;
  76.         if (searchValue) searchValue.value = term.value;
  77.  
  78.         this.booleanAnd = val.booleanAnd;
  79.         return val;
  80.     },
  81.  
  82.     // searchscope - just forward to the searchattribute
  83.     get searchScope() {
  84.         if (this.searchattribute)
  85.           return this.searchattribute.searchScope;
  86.         return undefined;
  87.     },
  88.     set searchScope(val) {
  89.         var searchAttribute = this.searchattribute;
  90.         if (searchAttribute) searchAttribute.searchScope=val;
  91.         return val;
  92.     },
  93.  
  94.     saveId: function (element, slot) {
  95.         this[slot] = element.id;
  96.     },
  97.  
  98.     getElement: function (slot) {
  99.         return document.getElementById(this[slot]);
  100.     },
  101.  
  102.     // three well-defined properties:
  103.     // searchattribute, searchoperator, searchvalue
  104.     // the trick going on here is that we're storing the Element's Id,
  105.     // not the element itself, because the XBL object may change out
  106.     // from underneath us
  107.     get searchattribute() { return this.getElement("internalSearchAttributeId"); },
  108.     set searchattribute(val) {
  109.         this.saveId(val, "internalSearchAttributeId");
  110.         return val;
  111.     },
  112.     get searchoperator() { return this.getElement("internalSearchOperatorId"); },
  113.     set searchoperator(val) {
  114.         this.saveId(val, "internalSearchOperatorId");
  115.         return val;
  116.     },
  117.     get searchvalue() { return this.getElement("internalSearchValueId"); },
  118.     set searchvalue(val) {
  119.         this.saveId(val, "internalSearchValueId");
  120.         return val;
  121.     },
  122.  
  123.     booleanNodes: null,
  124.     stringBundle: document.getElementById("bundle_search"),
  125.     get booleanAnd() { return this.internalBooleanAnd; },
  126.     set booleanAnd(val) {
  127.         this.internalBooleanAnd = val;
  128.         return val;
  129.     },
  130.  
  131.     save: function () {
  132.         var searchTerm = this.searchTerm;
  133.  
  134.         searchTerm.attrib = this.searchattribute.value;
  135.         var nsMsgSearchAttrib = Components.interfaces.nsMsgSearchAttrib;
  136.         if (this.searchattribute.value > nsMsgSearchAttrib.OtherHeader && this.searchattribute.value < nsMsgSearchAttrib.kNumMsgSearchAttributes) 
  137.           searchTerm.arbitraryHeader = this.searchattribute.label;
  138.         searchTerm.op = this.searchoperator.value;
  139.         if (this.searchvalue.value)
  140.             this.searchvalue.save();
  141.         else
  142.             this.searchvalue.saveTo(searchTerm.value);
  143.         searchTerm.value = this.searchvalue.value;
  144.         searchTerm.booleanAnd = this.booleanAnd;
  145.     },
  146.     // if you have a search term element with no search term
  147.     saveTo: function(searchTerm) {
  148.         this.internalSearchTerm = searchTerm;
  149.         this.save();
  150.     }
  151. }
  152.  
  153. var nsIMsgSearchTerm = Components.interfaces.nsIMsgSearchTerm;
  154.  
  155. function initializeSearchWidgets() 
  156. {    
  157.     gSearchBooleanRadiogroup = document.getElementById("booleanAndGroup");
  158.     gSearchTermList = document.getElementById("searchTermList");
  159.  
  160.     // initialize some strings
  161.     var bundle = document.getElementById('bundle_search');
  162.     gMoreButtonTooltipText = bundle.getString('moreButtonTooltipText');
  163.     gLessButtonTooltipText = bundle.getString('lessButtonTooltipText');
  164. }
  165.  
  166. function initializeBooleanWidgets() 
  167. {
  168.     var booleanAnd = true;
  169.     // get the boolean value from the first term
  170.     var firstTerm = gSearchTerms[0].searchTerm;
  171.     if (firstTerm)
  172.         booleanAnd = firstTerm.booleanAnd;
  173.  
  174.     // target radio items have value="and" or value="or"
  175.     gSearchBooleanRadiogroup.value = booleanAnd ? "and" : "or";
  176. }
  177.  
  178. function initializeSearchRows(scope, searchTerms)
  179. {
  180.     for (var i = 0; i < searchTerms.Count(); i++) {
  181.         var searchTerm = searchTerms.QueryElementAt(i, nsIMsgSearchTerm);
  182.         createSearchRow(i, scope, searchTerm);
  183.         gTotalSearchTerms++;
  184.     }
  185.     initializeBooleanWidgets();
  186.     updateRemoveRowButton();
  187. }
  188.  
  189. // enables/disables the less button for the first row of search terms.
  190. function updateRemoveRowButton()
  191. {
  192.   var firstListItem = gSearchTermList.getItemAtIndex(0);
  193.   if (firstListItem)
  194.     firstListItem.lastChild.lastChild.lastChild.setAttribute("disabled", gTotalSearchTerms == 1);
  195. }
  196.  
  197. // Returns the actual list item row index in the list of search rows
  198. // that contains the passed in element id.
  199. function getSearchRowIndexForElement(aElement)
  200. {
  201.   var listItem = aElement;
  202.   
  203.   while (listItem && listItem.localName != "listitem")
  204.     listItem = listItem.parentNode;
  205.     
  206.   return gSearchTermList.getIndexOfItem(listItem);
  207. }
  208.  
  209. function onMore(event)
  210. {
  211.   // if we have an event, extract the list row index and use that as the row number
  212.   // for our insertion point. If there is no event, append to the end....
  213.   var rowIndex; 
  214.  
  215.   if (event)
  216.     rowIndex = getSearchRowIndexForElement(event.target) + 1;
  217.   else
  218.     rowIndex = gSearchTermList.getRowCount();
  219.  
  220.   createSearchRow(rowIndex, gSearchScope, null);
  221.   gTotalSearchTerms++;
  222.   updateRemoveRowButton();
  223.  
  224.   // the user just added a term, so scroll to it
  225.   gSearchTermList.ensureIndexIsVisible(rowIndex);
  226. }
  227.  
  228. function onLess(event)
  229. {
  230.   if (event && gTotalSearchTerms > 1) 
  231.   {
  232.     removeSearchRow(getSearchRowIndexForElement(event.target));    
  233.     --gTotalSearchTerms;
  234.   }
  235.  
  236.   updateRemoveRowButton();
  237. }
  238.  
  239. // set scope on all visible searchattribute tags
  240. function setSearchScope(scope) 
  241. {
  242.     gSearchScope = scope;
  243.     for (var i=0; i<gSearchTerms.length; i++) {
  244.         gSearchTerms[i].obj.searchattribute.searchScope = scope;
  245.         gSearchTerms[i].scope = scope;
  246.         // act like the user "selected" this, see bug #202848
  247.         gSearchTerms[i].obj.searchattribute.onSelect(null /* no event */);  
  248.     }
  249. }
  250.  
  251. function updateSearchAttributes()
  252. {
  253.     for (var i=0; i<gSearchTerms.length; i++) 
  254.         gSearchTerms[i].obj.searchattribute.refreshList();
  255.     }
  256.  
  257. function booleanChanged(event) {
  258.     // when boolean changes, we have to update all the attributes on the search terms
  259.     var newBoolValue = (event.target.getAttribute("value") == "and") ? true : false;
  260.     if (document.getElementById("abPopup")) {
  261.       var selectedAB = document.getElementById("abPopup").selectedItem.id;
  262.       setSearchScope(GetScopeForDirectoryURI(selectedAB));
  263.     }
  264.     for (var i=0; i<gSearchTerms.length; i++) {
  265.         var searchTerm = gSearchTerms[i].obj;
  266.         searchTerm.booleanAnd = newBoolValue;
  267.     }
  268. }
  269.  
  270. function createSearchRow(index, scope, searchTerm)
  271. {
  272.     var searchAttr = document.createElement("searchattribute");
  273.     var searchOp = document.createElement("searchoperator");
  274.     var searchVal = document.createElement("searchvalue");
  275.     var enclosingBox = document.createElement('vbox');
  276.  
  277.     var buttonBox = document.createElement("hbox");
  278.     buttonBox.setAttribute("align", "start");
  279.     var moreButton = document.createElement("button");
  280.     var lessButton = document.createElement("button");
  281.     moreButton.setAttribute("class", "small-button");
  282.     moreButton.setAttribute("oncommand", "onMore(event);");
  283.     moreButton.setAttribute('label', '+');
  284.     moreButton.setAttribute('tooltiptext', gMoreButtonTooltipText);
  285.     lessButton.setAttribute("class", "small-button");
  286.     lessButton.setAttribute("oncommand", "onLess(event);");    
  287.     lessButton.setAttribute('label', '\u2212');
  288.     lessButton.setAttribute('tooltiptext', gLessButtonTooltipText);
  289.  
  290.     enclosingBox.setAttribute('align', 'right');
  291.  
  292.     // now set up ids:
  293.     searchAttr.id = "searchAttr" + gUniqueSearchTermCounter;
  294.     searchOp.id  = "searchOp" + gUniqueSearchTermCounter;
  295.     searchVal.id = "searchVal" + gUniqueSearchTermCounter;
  296.  
  297.     buttonBox.appendChild(moreButton);
  298.     buttonBox.appendChild(lessButton);
  299.  
  300.     searchAttr.setAttribute("for", searchOp.id + "," + searchVal.id);
  301.     searchOp.setAttribute("opfor", searchVal.id);
  302.  
  303.     var rowdata = new Array(enclosingBox, searchAttr,
  304.                             null, searchOp,
  305.                             null, searchVal,
  306.                             null, buttonBox);
  307.     var searchrow = constructRow(rowdata);
  308.     searchrow.id = "searchRow" + gUniqueSearchTermCounter;
  309.  
  310.     var searchTermObj = new searchTermContainer;
  311.     searchTermObj.searchattribute = searchAttr;
  312.     searchTermObj.searchoperator = searchOp;
  313.     searchTermObj.searchvalue = searchVal;
  314.  
  315.     // now insert the new search term into our list of terms
  316.     gSearchTerms.splice(index, 0, {obj:searchTermObj, scope:scope, searchTerm:searchTerm, initialized:false});
  317.  
  318.     // and/or string handling:
  319.     // this is scary - basically we want to take every other
  320.     // listcell, (note the i+=2) which will be a text label,
  321.     // and set the searchTermObj's
  322.     // booleanNodes to that
  323.     var stringNodes = new Array;
  324.     var listcells = searchrow.childNodes;
  325.     var j=0;
  326.     for (var i=0; i<listcells.length; i+=2) {
  327.       stringNodes[j++] = listcells[i];
  328.  
  329.       // see bug #183994 for why these cells are hidden
  330.       listcells[i].hidden = true;
  331.     }
  332.     searchTermObj.booleanNodes = stringNodes;
  333.  
  334.     var editFilter = null;
  335.     try { editFilter = gFilter; } catch(e) { }
  336.  
  337.     var editMailView = null;
  338.     try { editMailView = gMailView; } catch(e) { }
  339.  
  340.     if ((!editFilter && !editMailView) ||
  341.         (editFilter && index == gTotalSearchTerms) ||
  342.         (editMailView && index == gTotalSearchTerms))
  343.       gLoading = false;
  344.  
  345.     // index is index of new row
  346.     // gTotalSearchTerms has not been updated yet
  347.     if (gLoading || index == gTotalSearchTerms) {
  348.       gSearchTermList.appendChild(searchrow);
  349.     }
  350.     else {
  351.       var currentItem = gSearchTermList.getItemAtIndex(index);
  352.       gSearchTermList.insertBefore(searchrow, currentItem);
  353.     }
  354.     
  355.     // bump our unique search term counter
  356.     gUniqueSearchTermCounter++;
  357. }
  358.  
  359. function initializeTermFromId(id)
  360. {
  361.   initializeTermFromIndex(getSearchRowIndexForElement(document.getElementById(id)));
  362. }
  363.  
  364. function initializeTermFromIndex(index)
  365. {
  366.     var searchTermObj = gSearchTerms[index].obj;
  367.  
  368.     searchTermObj.searchScope = gSearchTerms[index].scope;
  369.     // the search term will initialize the searchTerm element, including
  370.     // .booleanAnd
  371.     if (gSearchTerms[index].searchTerm)
  372.         searchTermObj.searchTerm = gSearchTerms[index].searchTerm;
  373.     // here, we don't have a searchTerm, so it's probably a new element -
  374.     // we'll initialize the .booleanAnd from the existing setting in
  375.     // the UI
  376.     else
  377.     {
  378.       searchTermObj.booleanAnd = (gSearchBooleanRadiogroup.value == "and");
  379.       if (index)
  380.       {
  381.         // if we weren't pre-initialized with a searchTerm then steal the search attribute from the 
  382.         // previous row.
  383.         searchTermObj.searchattribute.value =  gSearchTerms[index - 1].obj.searchattribute.value;
  384.       }
  385.     }
  386.  
  387.     gSearchTerms[index].initialized = true;
  388. }
  389.  
  390. // creates a <listitem> using the array children as
  391. // the children of each listcell
  392. function constructRow(children)
  393. {
  394.     var listitem = document.createElement("listitem");
  395.     listitem.setAttribute("allowevents", "true");
  396.     for (var i = 0; i < children.length; i++) {
  397.       var listcell = document.createElement("listcell");
  398.  
  399.       // it's ok to have empty cells
  400.       if (children[i]) {
  401.           children[i].setAttribute("flex", "1");
  402.           listcell.appendChild(children[i]);
  403.       }
  404.       listitem.appendChild(listcell);
  405.     }
  406.     return listitem;
  407. }
  408.  
  409. function removeSearchRow(index)
  410. {
  411.     var searchTermObj = gSearchTerms[index].obj;
  412.     if (!searchTermObj) {
  413.         return;
  414.     }
  415.  
  416.     // if it is an existing (but offscreen) term,
  417.     // make sure it is initialized before we remove it.
  418.     if (!gSearchTerms[index].searchTerm && !gSearchTerms[index].initialized)
  419.         initializeTermFromIndex(index);
  420.  
  421.     // need to remove row from list, so walk upwards from the
  422.     // searchattribute to find the first <listitem>
  423.     var listitem = searchTermObj.searchattribute;
  424.  
  425.     while (listitem) {
  426.         if (listitem.localName == "listitem") break;
  427.         listitem = listitem.parentNode;
  428.     }
  429.  
  430.     if (!listitem) {
  431.         dump("Error: couldn't find parent listitem!\n");
  432.         return;
  433.     }
  434.  
  435.  
  436.     if (searchTermObj.searchTerm) {
  437.         gSearchRemovedTerms[gSearchRemovedTerms.length] = searchTermObj.searchTerm;
  438.     } else {
  439.         //dump("That wasn't real. ignoring \n");
  440.     }
  441.  
  442.     listitem.parentNode.removeChild(listitem);
  443.     
  444.     // now remove the item from our list of terms
  445.     gSearchTerms.splice(index, 1); 
  446. }
  447.  
  448. // save the search terms from the UI back to the actual search terms
  449. // searchTerms: nsISupportsArray of terms
  450. // termOwner:   object which can contain and create the terms
  451. //              (will be unnecessary if we just make terms creatable
  452. //               via XPCOM)
  453. function saveSearchTerms(searchTerms, termOwner)
  454. {
  455.     var i;
  456.     for (i = 0; i<gSearchTerms.length; i++) {
  457.         try {
  458.             var searchTerm = gSearchTerms[i].obj.searchTerm;
  459.  
  460.             // the term might be an offscreen one we haven't initialized yet
  461.             // if so, don't bother saving it.
  462.             if (!searchTerm && !gSearchTerms[i].initialized) {
  463.                 // is an existing term, but not initialize, so skip saving
  464.                 continue;
  465.             }
  466.  
  467.             if (searchTerm)
  468.                 gSearchTerms[i].obj.save();
  469.             else {
  470.                 // need to create a new searchTerm, and somehow save it to that
  471.                 searchTerm = termOwner.createTerm();
  472.                 gSearchTerms[i].obj.saveTo(searchTerm);
  473.                 termOwner.appendTerm(searchTerm);
  474.             }
  475.         } catch (ex) {
  476.             dump("** Error saving element " + i + ": " + ex + "\n");
  477.         }
  478.     }
  479.  
  480.     // now remove the queued elements
  481.     for (i=0; i<gSearchRemovedTerms.length; i++) {
  482.         // this is so nasty, we have to iterate through
  483.         // because GetIndexOf is acting funny
  484.         var searchTermSupports =
  485.             gSearchRemovedTerms[i].QueryInterface(Components.interfaces.nsISupports);
  486.         searchTerms.RemoveElement(searchTermSupports);
  487.     }
  488.  
  489. }
  490.  
  491. function onReset(event)
  492. {
  493.     while (gTotalSearchTerms>0)
  494.         removeSearchRow(--gTotalSearchTerms);
  495.     onMore(null);
  496. }
  497.  
  498. // this is a helper routine used by our search term xbl widget
  499. var gLabelStrings = new Array;
  500. function GetLabelStrings()
  501. {
  502.   if (!gLabelStrings.length)
  503.   {
  504.     var prefString;
  505.     var pref = Components.classes["@mozilla.org/preferences-service;1"].getService(Components.interfaces.nsIPrefBranch);
  506.     for (var index = 0; index < 6; index++)
  507.     {
  508.       prefString = pref.getComplexValue("mailnews.labels.description." + index,  Components.interfaces.nsIPrefLocalizedString);
  509.       gLabelStrings[index] = prefString;
  510.     }
  511.   }
  512.   return gLabelStrings;
  513. }
  514.